home *** CD-ROM | disk | FTP | other *** search
/ Chip 2007 January, February, March & April / Chip-Cover-CD-2007-02.iso / Pakiet bezpieczenstwa / mini Pentoo LiveCD 2006.1 / mpentoo-2006.1.iso / livecd.squashfs / usr / lib / python2.4 / profile.pyo (.txt) < prev    next >
Python Compiled Bytecode  |  2005-10-18  |  16KB  |  499 lines

  1. # Source Generated with Decompyle++
  2. # File: in.pyo (Python 2.4)
  3.  
  4. '''Class for profiling Python code.'''
  5. import sys
  6. import os
  7. import time
  8. import marshal
  9. from optparse import OptionParser
  10. __all__ = [
  11.     'run',
  12.     'runctx',
  13.     'help',
  14.     'Profile']
  15.  
  16. def run(statement, filename = None, sort = -1):
  17.     '''Run statement under profiler optionally saving results in filename
  18.  
  19.     This function takes a single argument that can be passed to the
  20.     "exec" statement, and an optional file name.  In all cases this
  21.     routine attempts to "exec" its first argument and gather profiling
  22.     statistics from the execution. If no file name is present, then this
  23.     function automatically prints a simple profiling report, sorted by the
  24.     standard name string (file/line/function-name) that is presented in
  25.     each line.
  26.     '''
  27.     prof = Profile()
  28.     
  29.     try:
  30.         prof = prof.run(statement)
  31.     except SystemExit:
  32.         pass
  33.  
  34.     if filename is not None:
  35.         prof.dump_stats(filename)
  36.     else:
  37.         return prof.print_stats(sort)
  38.  
  39.  
  40. def runctx(statement, globals, locals, filename = None):
  41.     '''Run statement under profiler, supplying your own globals and locals,
  42.     optionally saving results in filename.
  43.  
  44.     statement and filename have the same semantics as profile.run
  45.     '''
  46.     prof = Profile()
  47.     
  48.     try:
  49.         prof = prof.runctx(statement, globals, locals)
  50.     except SystemExit:
  51.         pass
  52.  
  53.     if filename is not None:
  54.         prof.dump_stats(filename)
  55.     else:
  56.         return prof.print_stats()
  57.  
  58.  
  59. def help():
  60.     for dirname in sys.path:
  61.         fullname = os.path.join(dirname, 'profile.doc')
  62.         if os.path.exists(fullname):
  63.             sts = os.system('${PAGER-more} ' + fullname)
  64.             if sts:
  65.                 print '*** Pager exit status:', sts
  66.             
  67.             break
  68.             continue
  69.     else:
  70.         print 'Sorry, can\'t find the help file "profile.doc"', 'along the Python search path.'
  71.  
  72. if os.name == 'mac':
  73.     import MacOS
  74.     
  75.     def _get_time_mac(timer = MacOS.GetTicks):
  76.         return timer() / 60.0
  77.  
  78.  
  79. if hasattr(os, 'times'):
  80.     
  81.     def _get_time_times(timer = os.times):
  82.         t = timer()
  83.         return t[0] + t[1]
  84.  
  85.  
  86.  
  87. class Profile:
  88.     '''Profiler class.
  89.  
  90.     self.cur is always a tuple.  Each such tuple corresponds to a stack
  91.     frame that is currently active (self.cur[-2]).  The following are the
  92.     definitions of its members.  We use this external "parallel stack" to
  93.     avoid contaminating the program that we are profiling. (old profiler
  94.     used to write into the frames local dictionary!!) Derived classes
  95.     can change the definition of some entries, as long as they leave
  96.     [-2:] intact (frame and previous tuple).  In case an internal error is
  97.     detected, the -3 element is used as the function name.
  98.  
  99.     [ 0] = Time that needs to be charged to the parent frame\'s function.
  100.            It is used so that a function call will not have to access the
  101.            timing data for the parent frame.
  102.     [ 1] = Total time spent in this frame\'s function, excluding time in
  103.            subfunctions (this latter is tallied in cur[2]).
  104.     [ 2] = Total time spent in subfunctions, excluding time executing the
  105.            frame\'s function (this latter is tallied in cur[1]).
  106.     [-3] = Name of the function that corresponds to this frame.
  107.     [-2] = Actual frame that we correspond to (used to sync exception handling).
  108.     [-1] = Our parent 6-tuple (corresponds to frame.f_back).
  109.  
  110.     Timing data for each function is stored as a 5-tuple in the dictionary
  111.     self.timings[].  The index is always the name stored in self.cur[-3].
  112.     The following are the definitions of the members:
  113.  
  114.     [0] = The number of times this function was called, not counting direct
  115.           or indirect recursion,
  116.     [1] = Number of times this function appears on the stack, minus one
  117.     [2] = Total time spent internal to this function
  118.     [3] = Cumulative time that this function was present on the stack.  In
  119.           non-recursive functions, this is the total execution time from start
  120.           to finish of each invocation of a function, including time spent in
  121.           all subfunctions.
  122.     [4] = A dictionary indicating for each function name, the number of times
  123.           it was called by us.
  124.     '''
  125.     bias = 0
  126.     
  127.     def __init__(self, timer = None, bias = None):
  128.         self.timings = { }
  129.         self.cur = None
  130.         self.cmd = ''
  131.         self.c_func_name = ''
  132.         if bias is None:
  133.             bias = self.bias
  134.         
  135.         self.bias = bias
  136.         if timer is None:
  137.             if os.name == 'mac':
  138.                 self.timer = MacOS.GetTicks
  139.                 self.dispatcher = self.trace_dispatch_mac
  140.                 self.get_time = _get_time_mac
  141.             elif hasattr(time, 'clock'):
  142.                 self.timer = self.get_time = time.clock
  143.                 self.dispatcher = self.trace_dispatch_i
  144.             elif hasattr(os, 'times'):
  145.                 self.timer = os.times
  146.                 self.dispatcher = self.trace_dispatch
  147.                 self.get_time = _get_time_times
  148.             else:
  149.                 self.timer = self.get_time = time.time
  150.                 self.dispatcher = self.trace_dispatch_i
  151.         else:
  152.             self.timer = timer
  153.             t = self.timer()
  154.             
  155.             try:
  156.                 length = len(t)
  157.             except TypeError:
  158.                 self.get_time = timer
  159.                 self.dispatcher = self.trace_dispatch_i
  160.  
  161.             if length == 2:
  162.                 self.dispatcher = self.trace_dispatch
  163.             else:
  164.                 self.dispatcher = self.trace_dispatch_l
  165.             
  166.             def get_time_timer(timer = timer, sum = sum):
  167.                 return sum(timer())
  168.  
  169.             self.get_time = get_time_timer
  170.         self.t = self.get_time()
  171.         self.simulate_call('profiler')
  172.  
  173.     
  174.     def trace_dispatch(self, frame, event, arg):
  175.         timer = self.timer
  176.         t = timer()
  177.         t = t[0] + t[1] - self.t - self.bias
  178.         if event == 'c_call':
  179.             self.c_func_name = arg.__name__
  180.         
  181.         if self.dispatch[event](self, frame, t):
  182.             t = timer()
  183.             self.t = t[0] + t[1]
  184.         else:
  185.             r = timer()
  186.             self.t = r[0] + r[1] - t
  187.  
  188.     
  189.     def trace_dispatch_i(self, frame, event, arg):
  190.         timer = self.timer
  191.         t = timer() - self.t - self.bias
  192.         if event == 'c_call':
  193.             self.c_func_name = arg.__name__
  194.         
  195.         if self.dispatch[event](self, frame, t):
  196.             self.t = timer()
  197.         else:
  198.             self.t = timer() - t
  199.  
  200.     
  201.     def trace_dispatch_mac(self, frame, event, arg):
  202.         timer = self.timer
  203.         t = timer() / 60.0 - self.t - self.bias
  204.         if event == 'c_call':
  205.             self.c_func_name = arg.__name__
  206.         
  207.         if self.dispatch[event](self, frame, t):
  208.             self.t = timer() / 60.0
  209.         else:
  210.             self.t = timer() / 60.0 - t
  211.  
  212.     
  213.     def trace_dispatch_l(self, frame, event, arg):
  214.         get_time = self.get_time
  215.         t = get_time() - self.t - self.bias
  216.         if event == 'c_call':
  217.             self.c_func_name = arg.__name__
  218.         
  219.         if self.dispatch[event](self, frame, t):
  220.             self.t = get_time()
  221.         else:
  222.             self.t = get_time() - t
  223.  
  224.     
  225.     def trace_dispatch_exception(self, frame, t):
  226.         (rpt, rit, ret, rfn, rframe, rcur) = self.cur
  227.         if rframe is not frame and rcur:
  228.             return self.trace_dispatch_return(rframe, t)
  229.         
  230.         self.cur = (rpt, rit + t, ret, rfn, rframe, rcur)
  231.         return 1
  232.  
  233.     
  234.     def trace_dispatch_call(self, frame, t):
  235.         if self.cur and frame.f_back is not self.cur[-2]:
  236.             (rpt, rit, ret, rfn, rframe, rcur) = self.cur
  237.             if not isinstance(rframe, Profile.fake_frame):
  238.                 self.trace_dispatch_return(rframe, 0)
  239.             
  240.         
  241.         fcode = frame.f_code
  242.         fn = (fcode.co_filename, fcode.co_firstlineno, fcode.co_name)
  243.         self.cur = (t, 0, 0, fn, frame, self.cur)
  244.         timings = self.timings
  245.         if fn in timings:
  246.             (cc, ns, tt, ct, callers) = timings[fn]
  247.             timings[fn] = (cc, ns + 1, tt, ct, callers)
  248.         else:
  249.             timings[fn] = (0, 0, 0, 0, { })
  250.         return 1
  251.  
  252.     
  253.     def trace_dispatch_c_call(self, frame, t):
  254.         fn = ('', 0, self.c_func_name)
  255.         self.cur = (t, 0, 0, fn, frame, self.cur)
  256.         timings = self.timings
  257.         if timings.has_key(fn):
  258.             (cc, ns, tt, ct, callers) = timings[fn]
  259.             timings[fn] = (cc, ns + 1, tt, ct, callers)
  260.         else:
  261.             timings[fn] = (0, 0, 0, 0, { })
  262.         return 1
  263.  
  264.     
  265.     def trace_dispatch_return(self, frame, t):
  266.         if frame is not self.cur[-2]:
  267.             self.trace_dispatch_return(self.cur[-2], 0)
  268.         
  269.         (rpt, rit, ret, rfn, frame, rcur) = self.cur
  270.         rit = rit + t
  271.         frame_total = rit + ret
  272.         (ppt, pit, pet, pfn, pframe, pcur) = rcur
  273.         self.cur = (ppt, pit + rpt, pet + frame_total, pfn, pframe, pcur)
  274.         timings = self.timings
  275.         (cc, ns, tt, ct, callers) = timings[rfn]
  276.         if not ns:
  277.             ct = ct + frame_total
  278.             cc = cc + 1
  279.         
  280.         if pfn in callers:
  281.             callers[pfn] = callers[pfn] + 1
  282.         else:
  283.             callers[pfn] = 1
  284.         timings[rfn] = (cc, ns - 1, tt + rit, ct, callers)
  285.         return 1
  286.  
  287.     dispatch = {
  288.         'call': trace_dispatch_call,
  289.         'exception': trace_dispatch_exception,
  290.         'return': trace_dispatch_return,
  291.         'c_call': trace_dispatch_c_call,
  292.         'c_exception': trace_dispatch_return,
  293.         'c_return': trace_dispatch_return }
  294.     
  295.     def set_cmd(self, cmd):
  296.         if self.cur[-1]:
  297.             return None
  298.         
  299.         self.cmd = cmd
  300.         self.simulate_call(cmd)
  301.  
  302.     
  303.     class fake_code:
  304.         
  305.         def __init__(self, filename, line, name):
  306.             self.co_filename = filename
  307.             self.co_line = line
  308.             self.co_name = name
  309.             self.co_firstlineno = 0
  310.  
  311.         
  312.         def __repr__(self):
  313.             return repr((self.co_filename, self.co_line, self.co_name))
  314.  
  315.  
  316.     
  317.     class fake_frame:
  318.         
  319.         def __init__(self, code, prior):
  320.             self.f_code = code
  321.             self.f_back = prior
  322.  
  323.  
  324.     
  325.     def simulate_call(self, name):
  326.         code = self.fake_code('profile', 0, name)
  327.         if self.cur:
  328.             pframe = self.cur[-2]
  329.         else:
  330.             pframe = None
  331.         frame = self.fake_frame(code, pframe)
  332.         self.dispatch['call'](self, frame, 0)
  333.  
  334.     
  335.     def simulate_cmd_complete(self):
  336.         get_time = self.get_time
  337.         t = get_time() - self.t
  338.         while self.cur[-1]:
  339.             self.dispatch['return'](self, self.cur[-2], t)
  340.             t = 0
  341.         self.t = get_time() - t
  342.  
  343.     
  344.     def print_stats(self, sort = -1):
  345.         import pstats as pstats
  346.         pstats.Stats(self).strip_dirs().sort_stats(sort).print_stats()
  347.  
  348.     
  349.     def dump_stats(self, file):
  350.         f = open(file, 'wb')
  351.         self.create_stats()
  352.         marshal.dump(self.stats, f)
  353.         f.close()
  354.  
  355.     
  356.     def create_stats(self):
  357.         self.simulate_cmd_complete()
  358.         self.snapshot_stats()
  359.  
  360.     
  361.     def snapshot_stats(self):
  362.         self.stats = { }
  363.         for cc, ns, tt, ct, callers in self.timings.iteritems():
  364.             callers = callers.copy()
  365.             nc = 0
  366.             for callcnt in callers.itervalues():
  367.                 nc += callcnt
  368.             
  369.             self.stats[func] = (cc, nc, tt, ct, callers)
  370.         
  371.  
  372.     
  373.     def run(self, cmd):
  374.         import __main__ as __main__
  375.         dict = __main__.__dict__
  376.         return self.runctx(cmd, dict, dict)
  377.  
  378.     
  379.     def runctx(self, cmd, globals, locals):
  380.         self.set_cmd(cmd)
  381.         sys.setprofile(self.dispatcher)
  382.         
  383.         try:
  384.             exec cmd in globals, locals
  385.         finally:
  386.             sys.setprofile(None)
  387.  
  388.         return self
  389.  
  390.     
  391.     def runcall(self, func, *args, **kw):
  392.         self.set_cmd(repr(func))
  393.         sys.setprofile(self.dispatcher)
  394.         
  395.         try:
  396.             return func(*args, **kw)
  397.         finally:
  398.             sys.setprofile(None)
  399.  
  400.  
  401.     
  402.     def calibrate(self, m, verbose = 0):
  403.         if self.__class__ is not Profile:
  404.             raise TypeError('Subclasses must override .calibrate().')
  405.         
  406.         saved_bias = self.bias
  407.         self.bias = 0
  408.         
  409.         try:
  410.             return self._calibrate_inner(m, verbose)
  411.         finally:
  412.             self.bias = saved_bias
  413.  
  414.  
  415.     
  416.     def _calibrate_inner(self, m, verbose):
  417.         get_time = self.get_time
  418.         
  419.         def f1(n):
  420.             for i in range(n):
  421.                 x = 1
  422.             
  423.  
  424.         
  425.         def f(m, f1 = f1):
  426.             for i in range(m):
  427.                 f1(100)
  428.             
  429.  
  430.         f(m)
  431.         t0 = get_time()
  432.         f(m)
  433.         t1 = get_time()
  434.         elapsed_noprofile = t1 - t0
  435.         if verbose:
  436.             print 'elapsed time without profiling =', elapsed_noprofile
  437.         
  438.         p = Profile()
  439.         t0 = get_time()
  440.         p.runctx('f(m)', globals(), locals())
  441.         t1 = get_time()
  442.         elapsed_profile = t1 - t0
  443.         if verbose:
  444.             print 'elapsed time with profiling =', elapsed_profile
  445.         
  446.         total_calls = 0.0
  447.         reported_time = 0.0
  448.         for filename, line, funcname in p.timings.items():
  449.             (cc, ns, tt, ct, callers) = None
  450.             if funcname in ('f', 'f1'):
  451.                 total_calls += cc
  452.                 reported_time += tt
  453.                 continue
  454.         
  455.         if verbose:
  456.             print "'CPU seconds' profiler reported =", reported_time
  457.             print 'total # calls =', total_calls
  458.         
  459.         if total_calls != m + 1:
  460.             raise ValueError('internal error: total calls = %d' % total_calls)
  461.         
  462.         mean = (reported_time - elapsed_noprofile) / 2.0 / total_calls
  463.         if verbose:
  464.             print 'mean stopwatch overhead per profile event =', mean
  465.         
  466.         return mean
  467.  
  468.  
  469.  
  470. def Stats(*args):
  471.     print 'Report generating functions are in the "pstats" module\x07'
  472.  
  473. if __name__ == '__main__':
  474.     usage = 'profile.py [-o output_file_path] [-s sort] scriptfile [arg] ...'
  475.     if not sys.argv[1:]:
  476.         print 'Usage: ', usage
  477.         sys.exit(2)
  478.     
  479.     
  480.     class ProfileParser(OptionParser):
  481.         
  482.         def __init__(self, usage):
  483.             OptionParser.__init__(self)
  484.             self.usage = usage
  485.  
  486.  
  487.     parser = ProfileParser(usage)
  488.     parser.allow_interspersed_args = False
  489.     parser.add_option('-o', '--outfile', dest = 'outfile', help = 'Save stats to <outfile>', default = None)
  490.     parser.add_option('-s', '--sort', dest = 'sort', help = 'Sort order when printing to stdout, based on pstats.Stats class', default = -1)
  491.     (options, args) = parser.parse_args()
  492.     sys.argv[:] = args
  493.     if len(sys.argv) > 0:
  494.         sys.path.insert(0, os.path.dirname(sys.argv[0]))
  495.         run('execfile(%r)' % (sys.argv[0],), options.outfile, options.sort)
  496.     else:
  497.         print 'Usage: ', usage
  498.  
  499.